
// Compiler implementation of the D programming language
// Copyright (c) 1999-2007 by Digital Mars
// All Rights Reserved
// written by Walter Bright
// http://www.digitalmars.com
// License for redistribution is by either the Artistic License
// in artistic.txt, or the GNU General Public License in gnu.txt.
// See the included readme.txt for details.

#include <stdio.h>
#include <string.h>
#include <ctype.h>
#include <assert.h>

#include "../l_service/c_ds/root.h"

#include "init.h"
#include "declaration.h"
#include "aggregate.h"
#include "mtype.h"
#include "attrib.h"
#include "template.h"
#include "../l_neo_lexer/id.h"
#include "module.h"

#if TARGET_LINUX || TARGET_OSX
char *cpp_mangle(P_Dsymbol *s);
#endif

char *mangle(P_Declaration *sthis)
{
    OutBuffer buf;
    char *id;
    P_Dsymbol *s;

    //printf("::mangle(%s)\n", sthis->oToChars());
    s = sthis;
    do
    {
	//printf("mangle: s = %p, '%s', parent = %p\n", s, s->oToChars(), s->parent);
	if (s->ident)
	{
	    P_FuncDeclaration *fd = s->isFuncDeclaration();
	    if (s != sthis && fd)
	    {
		id = mangle(fd);
		buf.prependstring(id);
		goto L1;
	    }
	    else
	    {
		id = s->ident->oToChars();
		int len = strlen(id);
		char tmp[sizeof(len) * 3 + 1];
		buf.prependstring(id);
		sprintf(tmp, "%d", len);
		buf.prependstring(tmp);
	    }
	}
	else
	    buf.prependstring("0");
	s = s->parent;
    } while (s);

//    buf.prependstring("_D");
L1:
    //printf("deco = '%s'\n", sthis->type->deco ? sthis->type->deco : "null");
    //printf("sthis->type = %s\n", sthis->type->oToChars());
    P_FuncDeclaration *fd = sthis->isFuncDeclaration();
    if (fd && (fd->needThis() || fd->isNested()))
	buf.writeByte(Type::needThisPrefix());
    if (sthis->type->deco)
	buf.writestring(sthis->type->deco);
    else
    {	assert(fd->inferRetType);
    }

    id = buf.oToChars();
    buf.data = NULL;
    return id;
}

char *P_Declaration::mangle()
#if __DMC__
    __out(result)
    {
	int len = strlen(result);

	assert(len > 0);
	//printf("mangle: '%s' => '%s'\n", oToChars(), result);
	for (int i = 0; i < len; i++)
	{
	    assert(result[i] == '_' ||
		   result[i] == '@' ||
		   isalnum(result[i]) || result[i] & 0x80);
	}
    }
    __body
#endif
    {
	//printf("P_Declaration::mangle(this = %p, '%s', parent = '%s', linkage = %d)\n", this, oToChars(), parent ? parent->oToChars() : "null", linkage);
	if (!parent || parent->isModule() || linkage == LINKcpp) // if at global scope
	{
	    // If it's not a D declaration, no mangling
	    switch (linkage)
	    {
		case LINKd:
		    break;

		case LINKc:
		case LINKwindows:
		case LINKpascal:
		    return ident->oToChars();

		case LINKcpp:
#if V2 && (TARGET_LINUX || TARGET_OSX)
		    return cpp_mangle(this);
#else
		    // Windows C++ mangling is done by C++ back end
		    return ident->oToChars();
#endif

		case LINKdefault:
		    error("forward declaration");
		    return ident->oToChars();

		default:
		    fprintf(stdmsg, "'%s', linkage = %d\n", oToChars(), linkage);
		    assert(0);
	    }
	}
	char *p = ::mangle(this);
	OutBuffer buf;
	buf.writestring("_D");
	buf.writestring(p);
	p = buf.oToChars();
	buf.data = NULL;
	//printf("P_Declaration::mangle(this = %p, '%s', parent = '%s', linkage = %d) = %s\n", this, oToChars(), parent ? parent->oToChars() : "null", linkage, p);
	return p;
    }

char *P_FuncDeclaration::mangle()
#if __DMC__
    __out(result)
    {
	assert(strlen(result) > 0);
    }
    __body
#endif
    {
	if (isMain())
	    return (char *)"_Dmain";

	if (isWinMain() || isDllMain() || ident == Id::tls_get_addr)
	    return ident->oToChars();

	assert(this);
	return P_Declaration::mangle();
    }

char *P_StructDeclaration::mangle()
{
    //printf("P_StructDeclaration::mangle() '%s'\n", oToChars());
    return P_Dsymbol::mangle();
}


char *P_TypedefDeclaration::mangle()
{
    //printf("P_TypedefDeclaration::mangle() '%s'\n", oToChars());
    return P_Dsymbol::mangle();
}


char *P_ClassDeclaration::mangle()
{
    P_Dsymbol *parentsave = parent;

    //printf("P_ClassDeclaration::mangle() %s.%s\n", parent->oToChars(), oToChars());

    /* These are reserved to the compiler, so keep simple
     * names for them.
     */
    if (ident == Id::Exception)
    {	if (parent->ident == Id::object)
	    parent = NULL;
    }
    else if (ident == Id::TypeInfo   ||
//	ident == Id::Exception ||
	ident == Id::TypeInfo_Struct   ||
	ident == Id::TypeInfo_Class    ||
	ident == Id::TypeInfo_Typedef  ||
	ident == Id::TypeInfo_Tuple ||
	this == object     ||
	this == classinfo  ||
	this == P_Module::moduleinfo ||
	memcmp(ident->oToChars(), "TypeInfo_", 9) == 0
       )
	parent = NULL;

    char *id = P_Dsymbol::mangle();
    parent = parentsave;
    return id;
}


char *P_TemplateInstance::mangle()
{
    OutBuffer buf;
    char *id;

#if 0
    printf("P_TemplateInstance::mangle() %s", toChars());
    if (parent)
	printf("  parent = %s %s", parent->kind(), parent->toChars());
    printf("\n");
#endif
    id = ident ? ident->oToChars() : oToChars();
    if (tempdecl->parent)
    {
	char *p = tempdecl->parent->mangle();
	if (p[0] == '_' && p[1] == 'D')
	    p += 2;
	buf.writestring(p);
    }
    buf.printf("%zu%s", strlen(id), id);
    id = buf.oToChars();
    buf.data = NULL;
    //printf("P_TemplateInstance::mangle() %s = %s\n", oToChars(), id);
    return id;
}



char *P_Dsymbol::mangle()
{
    OutBuffer buf;
    char *id;

#if 0
    printf("P_Dsymbol::mangle() '%s'", toChars());
    if (parent)
	printf("  parent = %s %s", parent->kind(), parent->toChars());
    printf("\n");
#endif
    id = ident ? ident->oToChars() : oToChars();
    if (parent)
    {
	char *p = parent->mangle();
	if (p[0] == '_' && p[1] == 'D')
	    p += 2;
	buf.writestring(p);
    }
    buf.printf("%zu%s", strlen(id), id);
    id = buf.oToChars();
    buf.data = NULL;
    //printf("P_Dsymbol::mangle() %s = %s\n", oToChars(), id);
    return id;
}


